-
Notifications
You must be signed in to change notification settings - Fork 3
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
MetrologyNamespace (initial PR) #9
base: main
Are you sure you want to change the base?
Conversation
src/metrology_apis/__init__.py
Outdated
def asdimension(obj: str | D) -> D: ... | ||
|
||
@staticmethod | ||
def asunit(obj) -> U[D]: ... |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
def asunit(obj) -> U[D]: ... | |
def asunit(obj: str | U) -> U[D]: ... |
should this be type hinted?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
perhaps #11 to consider
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks really good!
Minor comments about the docstring formatting.
src/metrology_apis/__init__.py
Outdated
VT = TypeVar('VT') | ||
DT = TypeVar('DT', bound='Dimension') | ||
UT = TypeVar('UT', bound='Unit[DT]') | ||
|
||
@runtime_checkable | ||
class MetrologyNamespace[Q: Quantity[VT, UT, DT], V, U: Unit[DT], D: Dimension](Protocol): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I tried introducing some TypeVar
s, but basedmypy isn't happy:
src/metrology_apis/__init__.py:16:38: error: Type variable "metrology_apis.VT" is unbound [valid-type]
class MetrologyNamespace[Q: Quantity[VT, UT, DT], V, U: Unit[DT], D: Dimension](Protocol):
^
src/metrology_apis/__init__.py:16:38: note: (Hint: Use "Generic[VT]" or "Protocol[VT]" base class to bind "VT" inside a class)
src/metrology_apis/__init__.py:16:38: note: (Hint: Use "VT" in function signature to bind "VT" inside a function)
I don't understand what the hints are suggesting though. Any clue @jorenham ?
Co-authored-by: Nathaniel Starkman <[email protected]>
Co-authored-by: Nathaniel Starkman <[email protected]>
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I notice that there are several type parameters with generic upper bounds (basically everything that uses TypeVar
). As I mentioned previously, this is unfortunately not supported in Python.
I'd recommend adding type-checkers to the CI, so that you can avoid this in the future.
@jorenham yes, we have them, you can observe in CI the errors I pointed out at #9 (comment). My question is how to address those errors. |
In other words, do we have to use just |
Ah nice. I must've missed it because I did not expect type-checkers to run under the "lint" label, but it's of course true that type-checking is a form of linting.
Basedmypy is indeed the only one that supports it. But it requires that you use all off the type-parameters in the upper bound on the right-hand-side. Put differently; it doesn't accept free type-parameters. That's because free type-parameters don't do anything, so you might as well leave them out (and use However, by relying on this exclusive basedmypy feature, you're effectively requiring all of your users to only use basedmypy. Because, after all, the other type-checkers don't support these parametrized upper bounds of type-parameter. The typing-spec compliant alternative is precisely like you said — using |
Okay, thanks! How does b70a68b look? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
maybe I'm missing something, but I see only two type parameters here:
class Quantity[V, U: Unit](Protocol): |
Nitpick: Since you're using Quantity[Any, Any(, Any)?]
a couple of times now, you could consider defining a type _AnyQuantity = ...
for the the sake of DRY.
I think you're looking at a different commit to the tip of this branch |
okay I pushed some aliases. Looks like that has introduced 1 Mypy failure locally |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Overall, this looks nice, and the case for __metrology_namespace__
becomes quite clear.
What I'm confused about is the addition of the dimension to a Quantity
. Why is that there? This especially if the dimension is something that follows directly from the unit, not a QuantityType
. But even in the latter case, is that needed for a minimal API?
But perhaps I'm just misunderstanding how typing works; I also don't really understand why unit is changed to Unit[D: Dimension]
- does that at all imply a __class_getitem__
that takes dimension?
Having We are definitely into territory where it's helpful to have these things confirmed by those more knowledgeable with typing, though. |
|
Return MetrologyNamespace in docstirngs
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Adding the dimension type parameters seems like a good choice to me.
From a user perspective, it can become quite verbose if you always have to specify three required type-arguments for Quantity
. Two ways to help with that come to mind:
- Make them optional by setting PEP 696 type-parameter defaults. If use use
Any
as default inQuantity
, for example, then theQT
type alias won't be needed anymore. Implementation-wise, this requires usingtyping_extensions.TypeVar
, for compatibility with python <3.13, which isn't all that pretty unfortunately. - Re-parametrize
Quantity
using the entireMetrologyNamespace
instead. That way, all three type-parameters can be statically known by matching onself: Quantity[MetrologyNamespace[...]]
in methods. You're basically moving the problem here, though. Type-checker errors will also become more verbose this way. The additional advantage of this is that you now also know the exact (sub)type of theMetrologyNamespace
itself, including any additional (e.g. user-defined) methods and attributes.
we're actually requiring Python 3.13 right now for ease of development Line 4 in 05af87c
|
Excellent. Then defaults are basically free 👌 |
is there clear documentation for the syntax for defaults in Python 3.13? |
def asunit(obj: str | U) -> U: ... | ||
|
||
@staticmethod | ||
def asquantity(obj: Q | V, *, unit: U) -> Q: ... |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
could the unit be entered as a string?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
it could, but I don't think we should require that? At worst you would just need to do mn.asquantity(v, unit=mn.asunit('length')
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Agreed. 2 thoughts:
- The QAPI, like the AAPI, represents the minimum support required for an implementing library, so even if we only have
unit: U
implementing libraries can support strings. - Strings as inputs seem reasonable, but also something we can discuss later. Unit objects as inputs is definitely a must.
Yea, in PEP 696 for example |
@@ -32,7 +110,34 @@ def __rtruediv__(self, other: Self, /) -> Self: ... | |||
|
|||
|
|||
@runtime_checkable | |||
class Quantity[V, U: Unit](Protocol): | |||
class Quantity[V, U: UT, D: Dimension](Protocol): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
like this @jorenham ?
class Quantity[V, U: UT, D: Dimension](Protocol): | |
class Quantity[V = Any, U = Any, D = Any](Protocol): |
Bit larger question about I ask in part since this means one is combining two separate actions into one function, which I'm not totally sure is a good idea. This partially since it just immediately implies that the very first line in any implementation will be Now of course we do need a way to convert quantities, and/or more generally convert some
which would effectively be part of the units API (i.e., independent of
This second version could be used to convert a value via I'm not sure whether one should just have both (with different names, obviously), just have one, or combine, with |
I also feel that separation may be preferable. It does depart from the array API standard in the sense that def interpret_as_quantity(obj: Quantity[V, U] | V) -> Quantity[V, U]:
return obj if isinstance(obj, Quantity[V, U]) else mn.asquantity(obj) |
@@ -32,7 +110,34 @@ def __rtruediv__(self, other: Self, /) -> Self: ... | |||
|
|||
|
|||
@runtime_checkable | |||
class Quantity[V, U: Unit](Protocol): | |||
class Quantity[V, U: UT, D: Dimension](Protocol): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
you can keep the upper bounds:
class Quantity[V, U: UT, D: Dimension](Protocol): | |
class Quantity[V = Any, U: UT = Any, D: Dimension = Any](Protocol): |
|
||
|
||
@runtime_checkable | ||
class MetrologyNamespace[Q: QT, V: VT, U: UT, D: Dimension](Protocol): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
class MetrologyNamespace[Q: QT, V: VT, U: UT, D: Dimension](Protocol): | |
class MetrologyNamespace[Q: QT = Any, V: VT = Any, U: UT = Any, D: Dimension = Any](Protocol): |
or you could keep e.g. Q
van V
required:
class MetrologyNamespace[Q: QT, V: VT, U: UT, D: Dimension](Protocol): | |
class MetrologyNamespace[Q: QT, V: VT, U: UT = Any, D: Dimension = Any](Protocol): |
Ok. Rich set of comments from @mhvk and @lucascolley! Here's my 2¢, building off the comments From @mhvk
def uconvert(obj: Q[U], unit: U) -> Q[U]: ...
def uconvert_value(obj: V, from_unit: U, to_unit U) -> V: Then (this is not enforced by the API and would be discussed only in documentation) a very reasonable implementation of def uconvert(obj: Q[U], unit: U) -> Q[U]:
return asquantity(uconvert_value(obj.value, obj.unit, unit), unit) Which very nicely separates the concerns of a) converting the value in one unit to another and b) reconstituting the Quantity. From @mhvk
I see your perspectives and strongly sympathize. But I think I disagree. First, what I do 100% agree on is that def asquantity(value: V, unit: U) -> Q[V, U]: ... is the core operation of However, I do think that the similarity to @overload
def asquantity(value: V, unit: U) -> Q[V, U]: ...
@overload
def asquantity(value: Q[V, U], unit: U) -> Q[V, U]: ... The overlap between For example, something which # ExampleLibrary.py (implementing and extending the QAPI)
def asquantity(value: V, unit: U, dtype: DType = None) -> Q[V, U]: ... |
Thanks @nstarman, that sounds pretty good to me! If all existing libraries would be happy with implementing |
I like the idea but the names not as much 😺 Note that this being on the metrology namespace, it is OK to have a more common name ("Namespaces are one honking great idea" and all that). Maybe Aside, |
Suggested I move this here from #16 List of relevant to conversion methods from the library I maintain.
Measurement|Quantity
Standalone methods
|
Thanks! I'll add astropy equivalents, though will restate that we really should try to do the minimum. That said, if every unit library implements something, perhaps that is a sign that it belongs in the absolute minimum...
Unit.is_equivalent(unit, equivalencies=...)`, i.e., same as above, but we allow specifying what constitutes convertible units (including functions to possibly do the actual conversion).
We don't have this but the closest is
Is this for C to F, etc.? My sense would be that such things need a different unit class that know how what their offset is. In astropy, we actually never got around to dealing with this properly (since thankfully almost nobody uses C or F), but we do something similar for magnitudes, decibels and other logarithmic units. It is perhaps relevant, since that is one argument for carrying the conversion method on the unit rather than as a function (maybe better post separately below). |
Further to the above, maybe some relevant points (though likely for v2!):
|
one of the key requirements for the library was to support power systems units, which look like per-unit MW or per-unit V. These are essentially dimensionless units but it is necessary to track the original unit as well so to know what it can legitimately be converted back to. In the computation this allows transformers to be mostly ignored in some computations. But we need a mechanism to get back to the original values as well. Hence the use of dedicated per-unit flag in the unit definition. I wouldn't really suggest trying to make this mechanism more broadly applicable as it is a rather niche use case, though we do also use for other things like strain for example and other types of ratio measurements. For C to F and R and a few other temperature units there is a separate chunk of code that knows about those and do the appropriate conversion. There is also an equation flag and overloaded bit codes that know of different types of operations and inverse operations for things like dB and bel, and a few other logarithmic type operations that are inherent in some units. There isn't a general datum shift mechanism though. |
Interesting. That sounds not dissimilar to our logarithmic units, where we also have to keep track of the reference unit (e.g., |
transferred from quantity-dev/quantity-api#5